home *** CD-ROM | disk | FTP | other *** search
/ Amiga Plus 2000 #4 / Amiga Plus CD - 2000 - No. 4.iso / Vollversion / CamD / development / examples / playmf / playmf.c < prev    next >
C/C++ Source or Header  |  2000-05-15  |  30KB  |  929 lines

  1. ;/* playmf.c - Execute me to compile me with SAS C 6.x
  2. SC playmf.c data=near nominc strmer streq nostkchk saveds ign=73
  3. Slink FROM LIB:c.o,playmf.o TO playmf LIBRARY LIB:SC.lib,LIB:Amiga.lib ND NOICONS
  4. quit
  5. */
  6.  
  7. /*=======================================================================
  8.  
  9. This file is a prototype score player which reads standard MIDI files,
  10. format type 0 or 1.  It conforms to the 1.0 specification for standard 
  11. MIDI files by Dave Oppenheim of Opcode Systems.  It uses both the CAMD and 
  12. RealTime libraries by Roger Dannenberg, Bill Barton, David Joiner, et al.
  13.  
  14. Design and coding by Dan Baker, Commodore Business Machines.
  15.  
  16. The tempo handling used by this program is a bit crude and is known to have 
  17. deficiencies.  It should not be used as an example of proper tempo handling.  
  18. This program also makes extensive use of globals and other techniques which 
  19. are convenient to the program's author.  Apologies to the reader.
  20. =======================================================================*/
  21.  
  22. /* System Include Files */
  23. #include <exec/types.h>
  24. #include <exec/memory.h>
  25. #include <libraries/dosextens.h>
  26.  
  27. /* CAMD MIDI Library Includes */
  28. #include <clib/camd_protos.h>
  29. #include <midi/camd.h>
  30. #include <midi/camdbase.h>
  31. #include <midi/mididefs.h>
  32. #include <pragmas/camd_pragmas.h>
  33.  
  34. /* CAMD Real Time Library Includes */
  35. #include <clib/realtime_protos.h>
  36. #include <libraries/realtime.h>
  37. #include <pragmas/realtime_pragmas.h>
  38.  
  39. /* System function prototypes */
  40. #include <clib/exec_protos.h>
  41. #include <clib/dos_protos.h>
  42.  
  43. /* Lattice Standard I/O */
  44. #include <stdlib.h>
  45. #include <stdio.h>
  46.  
  47.  
  48. /**********    debug macros     ***********/
  49. #define MYDEBUG  1
  50. void kprintf(UBYTE *fmt,...);
  51. void dprintf(UBYTE *fmt,...);
  52. #define DEBTIME 0
  53. #define bug printf
  54. #if MYDEBUG
  55. #define D(x) (x); if(DEBTIME>0) Delay(DEBTIME);
  56. #else
  57. #define D(x) ;
  58. #endif /* MYDEBUG */
  59. /********** end of debug macros **********/
  60.  
  61.  
  62. #define MINARGS 2
  63.  
  64. UBYTE *vers = "\0$VER: playmf 40.1";
  65. UBYTE *usage = "Usage: playmf midifilename";
  66.  
  67. #ifdef __SASC
  68. void __regargs __chkabort(void) {}      /* Disable SAS CTRL-C checking. */
  69. #else
  70. #ifdef LATTICE
  71. void chkabort(void) {}                  /* Disable LATTICE CTRL-C checking */
  72. #endif
  73. #endif
  74.  
  75. /*-------------------*/
  76. /*     Prototypes    */
  77. /*-------------------*/
  78. void kill(char *killstring);
  79. ULONG ComVarLen (UBYTE *value);
  80. UBYTE *DecodeEvent(UBYTE *,struct DecTrack *, ULONG);
  81. LONG transfer(struct DecTrack *,ULONG,LONG);
  82. ULONG changetempo(ULONG,ULONG,ULONG);
  83.  
  84. /*---------------------*/
  85. /* S M F  Header  File */
  86. /*---------------------*/
  87. /* Four-character IDentifier builder*/
  88. #define MakeID(a,b,c,d)  ( (LONG)(a)<<24L | (LONG)(b)<<16L | (c)<<8 | (d) )
  89. #define MThd MakeID('M','T','h','d')
  90. #define MTrk MakeID('M','T','r','k')
  91.  
  92. struct SMFHeader {LONG     ChunkID;  /* 4 ASCII characters */
  93.                   LONG     VarLeng;
  94.                   WORD     Format;
  95.                   UWORD    Ntrks;
  96.                   WORD     Division;
  97.                  };
  98.  
  99.  
  100. struct DecTrack { ULONG absdelta;   /* 32-bit delta */
  101.                   ULONG nexclock;   /* storage */
  102.                   UBYTE status;     /* status from file */
  103.                   UBYTE rstatus;    /* running status from track */
  104.                   UBYTE d1;         /* data byte 1 */
  105.                   UBYTE d2;         /* data byte 2 */
  106.                   ULONG absmlength; /* 32-bit absolute metalength */
  107.                   UBYTE *endmarker;
  108.                   UBYTE metatype;   /* meta event type */
  109.                   BOOL playable;
  110.                   UBYTE pad3;
  111.                 };
  112.  
  113. /*------------------*/
  114. /* MIDI Header File */
  115. /*------------------*/
  116. #define MAXTRAX 64L
  117. #define TWOxMAXTRAX 128L
  118. #define MIDIBUFSIZE 512L
  119.  
  120. /* Library Bases */
  121. struct Library  *CamdBase,
  122.                 *RealTimeBase;
  123.  
  124. /* Compiler glue: stub functions for camd.library */
  125. struct MidiNode *CreateMidi(Tag tag, ...)
  126. {    return CreateMidiA((struct TagItem *)&tag );
  127. }
  128.  
  129. BOOL SetMidiAttrs(struct MidiNode *mi, Tag tag, ...)
  130. {    return SetMidiAttrsA(mi, (struct TagItem *)&tag );
  131. }
  132.  
  133. struct MidiLink *AddMidiLink(struct MidiNode *mi, LONG type, Tag tag, ...)
  134. {    return AddMidiLinkA(mi, type, (struct TagItem *)&tag );
  135. }
  136.  
  137.  
  138. /* Compiler glue: stub functions for realtime.library */
  139. BOOL SetPlayerAttrs(struct Player *pi, Tag tag, ...)
  140. {    return SetPlayerAttrsA(pi, (struct TagItem *)&tag );
  141. }
  142.  
  143. struct Player *CreatePlayer(Tag tag, ...)
  144. {    return CreatePlayerA( (struct TagItem *)&tag );
  145. }
  146.  
  147. #define TSHIFT  4
  148.  
  149. /*--------------*/
  150. /*   Globals    */
  151. /*--------------*/
  152. struct MidiLink   *pMidiLink;
  153. struct MidiNode   *pMidiNode;
  154. struct Player      *pPlayer;
  155. BPTR               smfhandle;
  156. UBYTE             *smfdata,*ptrack[TWOxMAXTRAX],*pData,trackct;
  157. LONG               midiSignal,smfdatasize,tempo_offs;
  158. ULONG           fillclock[2];
  159. ULONG              oldclock,sizeDTrack,tfactor,initfactor,division,donecount;
  160. UBYTE             *pfillbuf[2],lastRSchan;
  161. BOOL          Playing;
  162.  
  163.  
  164. #define ALLNOTESOFFLEN    48
  165. UBYTE AllNotesOff[ALLNOTESOFFLEN] = {
  166. MS_Ctrl | 0,  MM_AllOff, 0,
  167. MS_Ctrl | 1,  MM_AllOff, 0,
  168. MS_Ctrl | 2,  MM_AllOff, 0,
  169. MS_Ctrl | 3,  MM_AllOff, 0,
  170. MS_Ctrl | 4,  MM_AllOff, 0,
  171. MS_Ctrl | 5,  MM_AllOff, 0,
  172. MS_Ctrl | 6,  MM_AllOff, 0,
  173. MS_Ctrl | 7,  MM_AllOff, 0,
  174. MS_Ctrl | 8,  MM_AllOff, 0,
  175. MS_Ctrl | 9,  MM_AllOff, 0,
  176. MS_Ctrl | 10, MM_AllOff, 0,
  177. MS_Ctrl | 11, MM_AllOff, 0,
  178. MS_Ctrl | 12, MM_AllOff, 0,
  179. MS_Ctrl | 13, MM_AllOff, 0,
  180. MS_Ctrl | 14, MM_AllOff, 0,
  181. MS_Ctrl | 15, MM_AllOff, 0
  182. };
  183.  
  184. /*-----------------------------*/
  185. /*     CODE  Starts  Here      */
  186. /*-----------------------------*/
  187.  
  188. ULONG
  189. ComVarLen (value)
  190. UBYTE *value;
  191. {
  192.    register ULONG newval;
  193.    register UBYTE x;
  194.  
  195.    x=0;newval=0L;
  196.    while(x<4)
  197.    {
  198.       newval <<= 7;
  199.       newval |=  *(value+x) & 0x7f;
  200.       if(*(value+x) < 0x80)x=4;
  201.       x++;
  202.    }
  203.    return(newval);
  204. }
  205.  
  206.  
  207. void
  208. main(int argc, char **argv)
  209. {
  210.    char              *smfname,iobuffer[14];
  211.    struct SMFHeader  *pSMFHeader;
  212.    UBYTE             *pbyte,x;
  213.    WORD               w;
  214.    LONG               rdcount,y,z,res;
  215.    BOOL               notdone,timerr;
  216.    ULONG              masterswitch,lowclock,
  217.                       ylength[2],wakeup;
  218.  
  219.    BYTE               oldpri; /* Priority to restore */
  220.    struct Task       *mt;     /* Pointer to this task */
  221.  
  222.    UBYTE fillbuf1[MIDIBUFSIZE]; /* These buffers hold the notes translated */
  223.    UBYTE fillbuf2[MIDIBUFSIZE]; /* from the midifile file for playback          */
  224.    struct DecTrack *pDTrack[MAXTRAX];
  225.  
  226.  
  227.    if(((argc)&&(argc<MINARGS))||((argc>1)&&(argv[argc-1][0]=='?')))
  228.     {
  229.     printf("%s\n",usage);
  230.     exit(RETURN_OK);
  231.     }
  232.  
  233.    oldclock=0L;
  234.    smfdata=NULL;
  235.    smfhandle=0;
  236.    pMidiLink=NULL;
  237.    pMidiNode=NULL;
  238.    pPlayer=NULL;
  239.    RealTimeBase=NULL;
  240.    CamdBase=NULL;
  241.    mt=NULL;
  242.    notdone=TRUE;
  243.    pfillbuf[0]=fillbuf1;
  244.    pfillbuf[1]=fillbuf2;
  245.    fillclock[0]=0L;
  246.    fillclock[1]=0L;
  247.    donecount=0L;
  248.    lastRSchan=0xf1; /* Status of $F1 is undefined in Standard MIDI file Spec */
  249.    tempo_offs = 0;
  250.    tfactor= 12 << TSHIFT;
  251.    Playing = FALSE;
  252.  
  253.    /*--------------------------------------*/
  254.    /* Open the CAMD and RealTime libraries */
  255.    /*--------------------------------------*/
  256.    CamdBase=OpenLibrary("camd.library",0L);
  257.    if(!CamdBase)
  258.       kill("Can't open CAMD MIDI Library\n");
  259.  
  260.    RealTimeBase=OpenLibrary("realtime.library",0L);
  261.    if(!RealTimeBase)
  262.       kill("Can't open CAMD Real Time Library\n");
  263.  
  264.    /*---------------*/
  265.    /* Open the File */
  266.    /*---------------*/
  267.    if(argv[1]==NULL) 
  268.        kill("No file name given.\n");
  269.    smfname=argv[1];
  270.    smfhandle= Open( (UBYTE *)smfname , MODE_OLDFILE );
  271.    if(smfhandle==0)
  272.        kill("Cannot open file.\n");
  273.  
  274.  
  275.    /*-------------------------------*/
  276.    /* Read the SMF Header ID/Length */
  277.    /*-------------------------------*/
  278.    rdcount = Read(smfhandle,iobuffer,14);
  279.    if(rdcount==-1) 
  280.        kill ("Bad file during read.\n");
  281.    if(rdcount<14)  
  282.        kill ("Not a midifile, too short.\n");
  283.  
  284.    /*-----------------*/
  285.    /* Evaluate Header */
  286.    /*-----------------*/
  287.    pSMFHeader=(struct SMFHeader *)iobuffer;
  288.    if (pSMFHeader->ChunkID != MThd ) 
  289.       kill("Midifile has unknown header ID.\n");
  290.  
  291.    if (pSMFHeader->VarLeng != 6 ) 
  292.       kill("Midifile has unknown header.\n");
  293.  
  294.    if (pSMFHeader->Format == 0 ) 
  295.       printf("Parsing midifile format type 0\n");
  296.    else if (pSMFHeader->Format == 1 ) 
  297.       printf("Parsing midifile format type 1\n");
  298.    else                          
  299.       kill("Can't parse this midifile type.\n");
  300.  
  301.  
  302.    if(pSMFHeader->Ntrks >MAXTRAX )
  303.       kill("Midifile has more than MAXTRAX tracks.\n");
  304.    else
  305.       printf("Midifile has %ld tracks\n",pSMFHeader->Ntrks);
  306.  
  307.  
  308.    /*--------------------*/
  309.    /* Evaluate time base */
  310.    /*--------------------*/
  311.    if (pSMFHeader->Division < 0)
  312.    {
  313.       /* Real time: find frame rate */
  314.       w=(pSMFHeader->Division >> 8) & 0x007f;
  315.       if(w!=24 && w!=25 && w!=29 && w!=30)
  316.          kill("Non-metrical time specified; not MIDI/SMPTE frame rate\n");
  317.       else 
  318.          kill("Non-metrical time; MIDI/SMPTE frame rate\n");
  319.  
  320.       /* Real-time: find seconds resolution */
  321.       w=pSMFHeader->Division & 0x007f;
  322.       if(w==4)
  323.          kill("Non-metrical time in quarter seconds (MTC resolution)\n");
  324.       else if(w==8 || w==10 || w==80)
  325.          kill("Non-metrical time in 1/nths of a second (bit resolution)\n");
  326.       else
  327.          kill("Non-metrical time in 1/nths of a second\n");
  328.    }
  329.    else
  330.    {
  331.       printf("Midifile has 1/%ldth quarter notes per delta tick\n",
  332.               pSMFHeader->Division);
  333.       division=(ULONG)pSMFHeader->Division;
  334.  
  335.       /* According to "Standrd MIDI Files 1.0", July 1988, page 5 */
  336.       /* para. 4: "...time signature is assumed to be 4/4 and the */
  337.       /*           tempo 120 beats per minute."                   */
  338.       tfactor=changetempo(500000L,tfactor,0L); /* One quarter note every half second */
  339.    }
  340.  
  341.  
  342.    /*--------------------------------------*/
  343.    /* Calculate size and Read rest of file */
  344.    /*--------------------------------------*/
  345.    y = Seek(smfhandle,0L,OFFSET_END);
  346.    if(y==-1)
  347.       kill("Midifile or drive problem\n");
  348.    z = Seek(smfhandle,y,OFFSET_BEGINNING);
  349.    if(z==-1)
  350.       kill("Midifile or drive problem\n");
  351.  
  352.    smfdatasize=z-y; /* CDTV demo disk uses 12000 here */
  353.  
  354.    smfdata=AllocMem(smfdatasize,MEMF_PUBLIC | MEMF_CLEAR);
  355.    if(smfdata==NULL) 
  356.       kill("No memory for Midifile read.\n");
  357.  
  358.    rdcount = Read(smfhandle,smfdata,smfdatasize);
  359.    if(rdcount==-1)          
  360.       kill ("Bad file during read.\n");
  361.  
  362.    /* Commented out for CD ROM demo */
  363.    if(rdcount<smfdatasize)  
  364.       kill ("Midifile too short.\n");
  365.  
  366.  
  367.    /*----------------------*/
  368.    /* Find the MIDI tracks */
  369.    /*----------------------*/
  370.    trackct=0;
  371.    pbyte=smfdata;
  372.  
  373.    while((pbyte-smfdata < smfdatasize) && (trackct < MAXTRAX))
  374.    {
  375.       if((*pbyte=='M')&&(*(pbyte+1)=='T')&&
  376.          (*(pbyte+2)=='r')&&(*(pbyte+3)=='k'))
  377.       {
  378.          /* Beginning of track */
  379.          ptrack[trackct]=pbyte+8;
  380.          /* End of track marker */
  381.          ptrack[MAXTRAX+trackct-1]=pbyte;
  382.          trackct++;
  383.          pbyte+=4;
  384.       }
  385.       else pbyte++;
  386.    }
  387.  
  388.    /* End of track marker */   
  389.    ptrack[MAXTRAX+trackct-1]=pbyte;
  390.  
  391.    if(trackct != pSMFHeader->Ntrks)
  392.       kill("Too many or missing tracks\n");
  393.    printf("There are %ld tracks in this Midifile.\n",trackct);
  394.  
  395.  
  396.    /*----------------------------------------------*/
  397.    /* Set up a MidiNode and a MidiLink.  Link the  */
  398.    /* node to the default "out.0" MidiCluster .    */
  399.    /*----------------------------------------------*/
  400.    pMidiNode=CreateMidi( MIDI_Name,    "PlayMF Player",
  401.                          MIDI_MsgQueue, 0L,     /* This is a send-only   */
  402.                          MIDI_SysExSize,0L,     /* MIDI node so no input */
  403.                          TAG_END);              /* buffers are needed.   */
  404.    if(!pMidiNode)
  405.       kill("No memory for MIDI Node\n");
  406.  
  407.    pMidiLink=AddMidiLink( pMidiNode, MLTYPE_Sender, 
  408.                                      MLINK_Comment,  "PlayMF Player Link",
  409.                                      MLINK_Parse,    TRUE,
  410.                                      MLINK_Location, "out.0",
  411.                                      TAG_END);
  412.    if(!pMidiLink)
  413.       kill("No memory for MIDI Link\n");
  414.  
  415.  
  416.    /*----------------------------------------*/
  417.    /* Set up DTrack structure for each track */
  418.    /*----------------------------------------*/
  419.    sizeDTrack=trackct*sizeof(struct DecTrack);
  420.    pData=AllocMem( sizeDTrack, MEMF_PUBLIC | MEMF_CLEAR );
  421.    if(!pData)
  422.       kill("No memory for work area...\n");
  423.  
  424.    for(x=0;x<trackct;x++)
  425.    {
  426.       pDTrack[x]=(struct DecTrack *)
  427.                  (x * sizeof(struct DecTrack) + pData);
  428.       /* add end marker */
  429.       pDTrack[x]->endmarker = ptrack[MAXTRAX+x];
  430.    }
  431.  
  432.  
  433.    /*------------------------------------------------*/  
  434.    /* Get events from track into DecTrack structures */
  435.    /*------------------------------------------------*/
  436.    y=0;
  437.    masterswitch=0L;
  438.    lowclock=0xffffffff;
  439.  
  440.    /* Initialize DecTrack structures */
  441.    for(x=0;x<trackct;x++)
  442.    {  
  443.       /* Takes a pointer to the delta of a raw <MTrk event>, a pointer   */
  444.       /* to a DecTrack decoding structure to store the decoded event and */ 
  445.       /* a switch that tells which of the two buffers to use.  Returns a */
  446.       /* pointer to the next raw <MTrk event> in the track or 0 if the   */
  447.       /* track is exhausted.                                             */
  448.       ptrack[x] = DecodeEvent( ptrack[x] , pDTrack[x] , masterswitch );
  449.       if(pDTrack[x]->nexclock < lowclock && ptrack[x])
  450.          /* Find the first event */
  451.          lowclock=pDTrack[x]->nexclock;
  452.    }
  453.  
  454.    /*-----------------------------------*/
  455.    /* Transfer first events to A buffer */
  456.    /*-----------------------------------*/
  457.    for(x=0;x<trackct;x++)
  458.    {
  459.       if((pDTrack[x]->nexclock==lowclock) && ptrack[x])
  460.       {
  461.          /* Transfer event to parse buffer and handle successor */
  462.          y=transfer(pDTrack[x],masterswitch,y);
  463.          z=1;
  464.          while(z==1)
  465.          {
  466.             ptrack[x]=DecodeEvent(ptrack[x],pDTrack[x],masterswitch);
  467.             /* Next delta is zero... */
  468.             if( !(pDTrack[x]->absdelta) && ptrack[x])
  469.             {
  470.                y=transfer(pDTrack[x],masterswitch,y);
  471.             }
  472.             else {z=0;}
  473.          }
  474.       }
  475.    }
  476.    ylength[masterswitch]=y;
  477.    fillclock[masterswitch]=(ULONG)(((tfactor*lowclock)+tempo_offs) >> TSHIFT);
  478.  
  479.  
  480.  
  481.    /*------------------------------------*/
  482.    /* Transfer second events to B buffer */
  483.    /*------------------------------------*/
  484.    y=0;
  485.    masterswitch=1L;
  486.    lowclock=0xffffffff;
  487.    for(x=0;x<trackct;x++)
  488.    {
  489.       if(pDTrack[x]->nexclock < lowclock && ptrack[x])
  490.          lowclock=pDTrack[x]->nexclock;
  491.    }
  492.  
  493.    for(x=0;x<trackct;x++)
  494.    {
  495.       if(pDTrack[x]->nexclock==lowclock && ptrack[x])
  496.       {
  497.          /* Transfer event to parse buffer and handle successor */
  498.          y=transfer(pDTrack[x],masterswitch,y);
  499.          z=1;
  500.          while(z==1)
  501.          {
  502.             ptrack[x]=DecodeEvent(ptrack[x],pDTrack[x],masterswitch);
  503.             /* Next delta is zero... */
  504.             if( !(pDTrack[x]->absdelta) && ptrack[x])
  505.             {
  506.                y=transfer(pDTrack[x],masterswitch,y);
  507.             }
  508.             else {z=0;}
  509.          }
  510.       }
  511.    }
  512.  
  513.    ylength[masterswitch]=y;
  514.    fillclock[masterswitch]=(ULONG)(((tfactor*lowclock)+tempo_offs) >> TSHIFT);
  515.  
  516.    /*-----------------------------------------------------*/
  517.    /* Priority Must Be Above Intuition and Graphics Stuff */
  518.    /*-----------------------------------------------------*/ 
  519.    mt=FindTask(NULL);
  520.    oldpri=SetTaskPri(mt,21);
  521.  
  522.  
  523.    /*---------------------------------------------------------------*/ 
  524.    /* Set up a Player and a Conductor to get timing information */
  525.    /*---------------------------------------------------------------*/ 
  526.    midiSignal = AllocSignal(-1L);
  527.    if(midiSignal==-1) 
  528.        kill("Couldn't allocate a signal bit\n");
  529.  
  530.    pPlayer=CreatePlayer(     PLAYER_Name,    "PlayMF Player",
  531.                              PLAYER_Conductor,  "PlayMF Conductor",
  532.                              PLAYER_AlarmSigTask, mt,
  533.                              PLAYER_AlarmSigBit,midiSignal,
  534.                              TAG_END);
  535.    if(!pPlayer)
  536.        kill("Can't create a Player\n");
  537.  
  538.    /*---------------------------------*/
  539.    /* Make sure the clock is stopped. */
  540.    /*---------------------------------*/
  541.    res = SetConductorState( pPlayer, CONDSTATE_STOPPED, 0L );
  542.    if(res!=0)
  543.        kill("Couldn't stop conductor\n");
  544.  
  545.    /*-------------------------------------------*/
  546.    /* Play the first batch of notes in Buffer A */
  547.    /*-------------------------------------------*/
  548.    if(ylength[masterswitch^1]!=0)
  549.       {
  550.       ParseMidi(pMidiLink,pfillbuf[masterswitch^1],
  551.                           ylength[masterswitch^1]);
  552.       }
  553.  
  554.    /*------------------------------------*/
  555.    /* and start the RealTime alarm clock */
  556.    /*------------------------------------*/
  557.    res = SetConductorState( pPlayer, CONDSTATE_RUNNING, 0L );
  558.    if(res!=0)
  559.        kill("Couldn't start conductor\n");
  560.  
  561.    /*---------------------*/
  562.    /* and set the alarm.  */
  563.    /*---------------------*/
  564.    timerr = SetPlayerAttrs( pPlayer, 
  565.                             PLAYER_AlarmTime, fillclock[masterswitch],
  566.                             PLAYER_Ready, TRUE,
  567.                             TAG_END);
  568.    if(!timerr)
  569.        kill("Couldn't set player attrs\n");
  570.  
  571.    Playing = TRUE;
  572.  
  573.    /*-----------------*/
  574.    /* MAIN EVENT LOOP */
  575.    /*-----------------*/   
  576.    while(donecount<trackct)
  577.    {
  578.       masterswitch ^= 1L;
  579.       y=0;
  580.       lowclock=0xffffffff;
  581.  
  582.       /*------------------------------------------------*/  
  583.       /* Get events from track into DecTrack structures */
  584.       /*------------------------------------------------*/
  585.       for(x=0;x<trackct;x++)
  586.       {
  587.          if((pDTrack[x]->nexclock < lowclock) && ptrack[x])
  588.             lowclock=pDTrack[x]->nexclock;
  589.       }
  590.  
  591.       /*-----------------------------------*/
  592.       /* Transfer events to current buffer */
  593.       /*-----------------------------------*/
  594.       for(x=0;x<trackct;x++)
  595.       {
  596.          if((pDTrack[x]->nexclock==lowclock) && ptrack[x])
  597.          {
  598.             /* Transfer event to parse buffer and handle successor */
  599.             y=transfer(pDTrack[x],masterswitch,y);
  600.             z=1;
  601.             while(z==1)
  602.             {
  603.                ptrack[x]=DecodeEvent(ptrack[x],pDTrack[x],masterswitch);
  604.                /* Next delta is zero... */
  605.                if( !(pDTrack[x]->absdelta) && ptrack[x] )
  606.                {
  607.                   y=transfer(pDTrack[x],masterswitch,y);
  608.                }
  609.                else {z=0;}
  610.             }
  611.          }
  612.       }
  613.  
  614.       ylength[masterswitch]=y;
  615.       fillclock[masterswitch]=(LONG)(((tfactor*lowclock)+tempo_offs)>>TSHIFT);
  616.  
  617.       /*---------------------------------------------------------------*/
  618.       /* Wait() for the CAMD alarm or a CTRL-C keypress from the user. */
  619.       /*---------------------------------------------------------------*/
  620.       if(timerr)
  621.          wakeup=Wait(1L<< midiSignal | SIGBREAKF_CTRL_C);
  622.  
  623.       if(wakeup & SIGBREAKF_CTRL_C)
  624.       {
  625.      if(!(wakeup & 1L<<midiSignal)) Wait(1L<< midiSignal | SIGBREAKF_CTRL_C);
  626.          /* Restore Priority */
  627.          if(mt!=NULL) SetTaskPri(mt,oldpri);
  628.          /* And Quit */
  629.          kill("User abort\n");
  630.       }
  631.  
  632.       /*-------------------------------*/
  633.       /* Start the next set of events  */
  634.       /*-------------------------------*/
  635.       if(ylength[masterswitch^1]!=0)
  636.          {
  637.          ParseMidi(pMidiLink,pfillbuf[masterswitch^1],
  638.                              ylength[masterswitch^1]);
  639.          }
  640.  
  641.       /*---------------------*/
  642.       /* and set the alarm.  */
  643.       /*---------------------*/
  644.       timerr = SetPlayerAttrs( pPlayer, 
  645.                                PLAYER_AlarmTime, fillclock[masterswitch],
  646.                                PLAYER_Ready, TRUE,
  647.                                TAG_END);
  648.       if(!timerr)
  649.           kill("Couldn't set player attrs 2\n");
  650.    }
  651.  
  652.    /*-----------------------------------*/
  653.    /* Finish off the last set of events */
  654.    /*-----------------------------------*/
  655.    masterswitch^=1L;
  656.  
  657.    if(timerr)
  658.       wakeup=Wait(1L<< midiSignal | SIGBREAKF_CTRL_C);
  659.  
  660.  
  661.    if(ylength[masterswitch^1]!=0)
  662.       {
  663.       ParseMidi(pMidiLink,pfillbuf[masterswitch^1],
  664.                        ylength[masterswitch^1]);
  665.       }
  666.  
  667.    /* Restore Priority */
  668.    if(mt!=NULL) SetTaskPri(mt,oldpri);
  669.  
  670.    kill("");
  671. }
  672.  
  673.  
  674. /*------------------------------*/
  675. /* Cleanup and return resources */
  676. /*------------------------------*/
  677. void
  678. kill(char *killstring)
  679. {
  680.    if(Playing)           ParseMidi(pMidiLink,AllNotesOff,ALLNOTESOFFLEN);
  681.    if(pPlayer)         DeletePlayer(pPlayer);
  682.    if(midiSignal != -1)FreeSignal(midiSignal);
  683.    if(pData)           FreeMem(pData,sizeDTrack);
  684.    if(pMidiLink)       RemoveMidiLink(pMidiLink);
  685.    if(pMidiNode)       DeleteMidi(pMidiNode);
  686.    if(RealTimeBase)    CloseLibrary(RealTimeBase);
  687.    if(CamdBase)        CloseLibrary(CamdBase);
  688.    if(smfhandle)       Close(smfhandle);
  689.    if(smfdata)         FreeMem(smfdata,smfdatasize);
  690.    if(*killstring)     printf(killstring);
  691.    exit(0);
  692. }
  693.  
  694.  
  695. /*--------------------------------------------------*/  
  696. /* Translate from raw track data to a decoded event */
  697. /*--------------------------------------------------*/  
  698. UBYTE *DecodeEvent(UBYTE *ptdata,struct DecTrack *pDTdata, ULONG deswitch)
  699. {
  700.    LONG status;
  701.    ULONG length;
  702.    BOOL skipit;
  703.  
  704.    pDTdata->absdelta = 0L;
  705.    pDTdata->playable = TRUE; /* Assume it's playble and not a meta-event */
  706.  
  707.    skipit=FALSE;
  708.    do
  709.    {
  710.       /* is this track all used up? */             
  711.       if( ptdata >= pDTdata->endmarker )
  712.       {
  713.          /* printf("Track done,...\n"); */
  714.          donecount++;
  715.          return(0L);
  716.       }
  717.       else /* there is more data to handle in the track */
  718.       {
  719.          /* Decode delta */
  720.          pDTdata->absdelta += ComVarLen(ptdata);
  721.          pDTdata->nexclock+= pDTdata->absdelta;
  722.  
  723.          /* Update pointer to event following delta */
  724.          while(*ptdata>127)
  725.             {
  726.             ptdata++;
  727.             }
  728.          ptdata++;
  729.  
  730.          if(*ptdata>127) /* Event with status ($80-$FF): decode new status */  
  731.          {
  732.             status=*ptdata;
  733.          
  734.             pDTdata->status=status;
  735.             pDTdata->rstatus=0;    /* No running status was used */
  736.      
  737.             ptdata++;
  738.             
  739.             if(status<240) /* Handle easy status $8x - $Ex */
  740.             {
  741.                skipit=FALSE;
  742.                pDTdata->d1 = *ptdata;
  743.                if(status<192 || status>223) /* $80-$BF, $E0-$EF: 2 data bytes */
  744.                {
  745.                   ptdata++;
  746.                   pDTdata->d2=*ptdata;
  747.                }
  748.                else pDTdata->d2=0;            /* $C0-$DF: 1 data byte */
  749.             }
  750.             else /* Status byte $Fx, system exclusive or meta events  */
  751.             {
  752.                skipit=TRUE;
  753.                
  754.                if(status==0xff)            /* It's a meta event ($ff) */
  755.                {
  756.                   pDTdata->metatype=*ptdata;
  757.               
  758.                   ptdata++; /* Now pointing at length byte */
  759.  
  760.                   if(pDTdata->metatype==81)
  761.                   {
  762.                      /* Tempo change event.  There are 60 milllion    */
  763.                      /* microseconds in a minute.  The lower 3 bytes  */ 
  764.                      /* pointed to by ptdata give the microseconds    */
  765.                      /* per MIDI quarter note. So, assuming a quarter */
  766.                      /* note gets the beat, this equation             */
  767.                      /*      60000000L /                              */
  768.                      /*          ( *((ULONG *)ptdata) & 0x00ffffff ) )*/
  769.                      /* gives beats per minute.                       */
  770.  
  771.                      tfactor = changetempo( *((ULONG *)ptdata) & 0x00ffffff,
  772.                 tfactor,
  773.                 pDTdata->nexclock - pDTdata->absdelta);
  774.  
  775.                      /* Tempo event is not playable.  This prevents the */
  776.                      /* event from being transferred to the play buffer */
  777.                      pDTdata->playable = FALSE; 
  778.  
  779.                      /* Even though this event can't be played, it     */
  780.                      /* takes some time and should not be skipped.     */
  781.                      skipit=FALSE;
  782.                   }
  783.                   length=ComVarLen(ptdata);
  784.                   pDTdata->absmlength=length;
  785.                   while(*ptdata>127)ptdata++;
  786.  
  787.                   ptdata+=length;
  788.                }
  789.                else if(status==0xf0 || status==0xf7) /* It's a sysex event */
  790.                {
  791.                   pDTdata->metatype=0xff;
  792.                   printf("Sysex event");
  793.                   length=ComVarLen(ptdata);
  794.                   pDTdata->absmlength=length;
  795.                   while(*ptdata>127)ptdata++;
  796.  
  797.                   ptdata+=length;
  798.                }
  799.                else        /* It's an unkown event type ($f1-$f6, $f8-$fe) */
  800.                {
  801.                   pDTdata->metatype=0xff;
  802.                   printf("Unknown event"); 
  803.                }
  804.             }
  805.          }
  806.          else /* Event without status ($00-$7F): use running status */
  807.          {
  808.             skipit=FALSE;
  809.             /* Running status data bytes */
  810.             status=pDTdata->status;
  811.             pDTdata->rstatus=status;
  812.  
  813.             if(status==0)
  814.                kill("Bad file, data bytes without initial status.\n");
  815.  
  816.             pDTdata->d1=*ptdata;
  817.  
  818.             if(status<192 || status>223) /* $80-$BF, $E0-$EF: 2 data bytes */
  819.             {
  820.                ptdata++;
  821.                pDTdata->d2=*ptdata;
  822.             }
  823.             else pDTdata->d2=0;         /* $C0-$DF: 1 data byte */
  824.          }
  825.          ptdata++;
  826.       }
  827.    }
  828.    while(skipit);
  829.  
  830.    return(ptdata);
  831. }
  832.  
  833.  
  834. /*------------------------------------------------------------*/
  835. /* Transfer the decoded event to the fill buffer for playback */
  836. /*------------------------------------------------------------*/
  837. LONG
  838. transfer(struct DecTrack *pDT,ULONG mswitch,LONG ylen)
  839. {
  840.    ULONG y;
  841.    y=(ULONG )ylen;
  842.  
  843.    if (pDT->playable)
  844.    {
  845.       if(pDT->rstatus == lastRSchan)
  846.       { 
  847.          /* Running status so just put the 2 data bytes in buffer */
  848.          *(pfillbuf[mswitch] + y)=pDT->d1;
  849.          y++;
  850.          *(pfillbuf[mswitch] + y)=pDT->d2;
  851.       }
  852.       else 
  853.       {
  854.          /* New status so store status and data bytes */
  855.          *(pfillbuf[mswitch] + y)=pDT->status;
  856.          y++;
  857.          *(pfillbuf[mswitch] + y)=pDT->d1;
  858.          if(pDT->status<192 || pDT->status>223)
  859.          {
  860.             y++;
  861.             *(pfillbuf[mswitch] + y)=pDT->d2;
  862.          }
  863.          lastRSchan=pDT->status;     
  864.       }
  865.       y++;
  866.    }
  867.    return((LONG)y);
  868. }
  869.  
  870.  
  871. /*-------------------------------------------------------------------------*/
  872. /* Handle the Change Tempo event.   With the realtime.library, the timing  */
  873. /* quantum is fixed at .83 milliseconds (1.2kHz).  This makes handling of  */
  874. /* Playmf tempo a bit rough.  Tempo is controlled through a ULONG integer  */
  875. /* named tfactor which is used to multiply the time deltas in the Playmf   */
  876. /* file.  We can't multiply the time deltas by a fractional amount so we   */
  877. /* will shift up before dividing down for tfactor, and shift down after    */
  878. /* using the tfactor later.  This in effect gives us fractional control.   */
  879. /* To deal with the fact that tempo changes would cause us to wait for     */
  880. /* inappropriate past or future times on upcoming notes, we calculate a    */
  881. /* tempo_offs time offset (also shifted up) for use in time calculations.  */
  882. /*-------------------------------------------------------------------------*/
  883. ULONG
  884. changetempo(ULONG ctbpm, ULONG oldtfactor, ULONG eventtime)
  885. {
  886.    ULONG newtfactor,oldtime,newtime;
  887.  
  888.    /* ctbpm is the new microseconds per midi quarter note                */
  889.    /* Division is the part of a quarter note represented by a delta of 1 */
  890.    /* So ctbpm/division is the new microseconds per delta                */
  891.    /* CAMD uses 1200 ticks/sec or 833 microseconds per tick              */
  892.    /* We will compute a value which is:                                  */
  893.    /* ((microseconds per delta) << TSHIFT ) / 833                        */
  894.    /* The TSHIFT will give us some fractional control, and will be       */
  895.    /*    shifted out later when the new tfactor) is used.                */
  896.  
  897.    newtfactor = ((ctbpm << TSHIFT) / division) / 833; /* new tfactor */
  898.  
  899.    if((eventtime)&&(Playing))
  900.     {
  901.     /* We know our midi event time relative to the start of the file.
  902.      * We can calculate what CAMD time that equals at our initial tempo,
  903.      * and also at the new tempo.  We will save this difference as
  904.      * tempo_offs, and use tempo_offs to adjust all main program
  905.      * midi time -> CAMD time calculations.
  906.      * In other words, we figure out what CAMD time we would have
  907.      * been at now, if we had been playing at the new tempo all along.
  908.      * Using that information, we can adjust all upcoming MIDI
  909.      * Alarm wakeup times up or down by the amount the new tempo
  910.      * tfactor would have thrown us off.  Note that tempo_offs is
  911.      * based on values shifted up by TSHIFT, and this will be shifted
  912.      * out later in the main program time calculations.
  913.      */
  914.     oldtime = initfactor * eventtime; 
  915.     newtime = newtfactor * eventtime;
  916.     tempo_offs = oldtime - newtime;
  917.     /*printf("playtime=%ld oldtime=%ld newtime=%ld tempo_offs = %ld\n",
  918.         playtime, oldtime>>TSHIFT,newtime>>TSHIFT,tempo_offs>>TSHIFT); */
  919.     }
  920.    else
  921.     {
  922.     initfactor = newtfactor;
  923.     }
  924.    /* printf("changetempo: ctbpm=%ld newtfactor=%ld\n", ctbpm,newtfactor); */
  925.    return(newtfactor);
  926. }
  927.  
  928.  
  929.